import datetime
import random

from flask import request, abort, current_app, make_response, jsonify, session

from info.models import User
from . import passport_blue
from info.utils.captcha.captcha import captcha
from info import redis_store, constants, db
from info import response_code
import json, re
from info.libs.yuntongxun.sms import CCP


@passport_blue.route('/logout', methods=['GET'])
def logout():
    '''退出登录'''
    #     本质清理session
    try:
        session.pop('user_id', None)
        session.pop('mobile', None)
        session.pop('nick_name', None)

        session.pop('is_admin', False)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=response_code.RET.DBERR, errmsg='退出登录失败')

    return jsonify(errno=response_code.RET.OK, errmsg='退出登录成功')


@passport_blue.route('/login', methods=['POST'])
def login():
    '''登录
    1.接收参数(手机号,密码明文)
    2.判断参数是否完整,手机号是否合法
    3.查询根据手机号查询用户信息
    4.效验验证码是否正确
    5.将状态保持写入session,完成登录
    6.记录最后一次的登录时间
    7.响应登录结果
    '''
    # 1
    json_dict = request.json
    mobile = json_dict.get('mobile')
    password = json_dict.get('password')
    #  2.效验参数

    if not all([mobile, password]):
        return jsonify(errno=response_code.RET.PARAMERR, errmsg='缺少参数')

    if not re.match(r'^1[345678][0-9]{9}$', mobile):
        return jsonify(errno=response_code.RET.DBERR, errmsg='手机号格式错误')
        #     3.查询用户信息
    try:
        user = User.query.filter(User.mobile == mobile).first()
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=response_code.RET.DBERR, errmsg='用户查询失败')
    if not user:
        return jsonify(errno=response_code.RET.PARAMERR, errmsg='用户或密码错误')
    # 4/效验密码是否正确
    if not user.check_password(password):
        return jsonify(errno=response_code.RET.PARAMERR, errmsg='用户或密码错误')
    #     5.
    session['user_id'] = user.id
    session['mobile'] = user.mobile
    session['nick_name'] = user.nick_name
    # 6
    user.last_login = datetime.datetime.now()
    try:
        db.session.commit()
    except Exception as e:
        current_app.logger.error(e)
        db.session.rollback()
        return jsonify(errno=response_code.RET.DBERR, errmsg='记录最后一次登录失败')
    # 7
    return jsonify(errno=response_code.RET.OK, errmsg='登录成功')


@passport_blue.route('/register', methods=['POST'])
def register():
    '''注册
    1.接受参数(手机号,验证码,mm)
    2.效验参数是否合法和缺少
    3.查询服务器的短信验证码
    4.跟客户端传入的短信验证码进行比较
    5.若果对比成功,创建user模型对象,并对属性赋值
    6.将模型对象储存在数据库
    7.保存session ,实现状态保持,注册即登录
    8.响应注册结果
    '''
    # 1.接收参数
    json_dict = request.json
    mobile = json_dict.get('mobile')
    smscode_client = json_dict.get('smscode')
    password = json_dict.get('password')
    # 2.效验判断是否缺少和手机是否合法
    if not all([mobile, smscode_client, password]):
        return jsonify(errno=response_code.RET.PARAMERR, errmsg='缺少参数')
    if not re.match(r'^1[345678][0-9]{9}$', mobile):
        return jsonify(errno=response_code.RET.PARAMERR, errmsg='手机号格式缺少参数')
    # 3.查询服务器的短信验证码
    try:
        smscode_server = redis_store.get('SMS:' + mobile)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=response_code.RET.DBERR, errmsg='查询短信验证码失败')
    if not smscode_server:
        return jsonify(errno=response_code.RET.NODATA, errmsg='短信验证码不存在')
    # 4.跟客户端进行比较
    if smscode_client != smscode_server:
        return jsonify(errno=response_code.RET.PARAMERR, errmsg='短信验证码输入有误')
    # 5.对比成功,创建User模型对象
    user = User()
    user.mobile = mobile
    user.nick_name = mobile
    #     TODO密码加密后在储存
    #     方案一:在这个视图中,直接调用对应的密码加密算法,加密密码直接存储到数据库
    #     方案二:在这个视图中,封装一个加密方法,加密密码储存到数据库中
    #     方案三:在模型类中新增一个属性叫做password,并加载setter和getter方法,调用setter和getter方法,调用setter方法,直接完成加密的加密储存
    #     psd = user.password  获取属性值
    user.password = password  # setter 方法
    user.last_login = datetime.datetime.now()
    # 6.将模型对象储存在数据库
    try:
        db.session.add(user)
        db.session.commit()
    except Exception as e:
        current_app.logger.error(e)
        db.session.rollback()
        return jsonify(errno=response_code.RET.DBERR, errmsg='保存注册数据失败')
    # 7.保存session ,实现状态保持,注册即登录
    session['user_id'] = user.id
    session['mobile'] = user.mobile
    session['nick_name'] = user.nick_name

    #     8.注册成功
    return jsonify(errno=response_code.RET.OK, errmsg='注册成功')


@passport_blue.route('/sms_code', methods=['POST'])
def sms_code():
    '''发送短信
    1.接受参数(手机号,图片验证码,uuid)
    2.效验参数是否齐全,手机号是否合法
    3.查询服务器储存的图片验证码
    4.跟客户端传入的图片验证码对比
    5.如果对比成功,生成短信验证码,并发送短信
    6.存储短信验证码到redis,方便注册时比较
    7.响应短信验证码发送的结果
    '''
    # 1.接受参数(手机号,图片验证码,uuid)
    json_str = request.data
    json_dict = json.loads(json_str)
    mobile = json_dict.get('mobile')
    image_code_client = json_dict.get('image_code')
    image_code_id = json_dict.get('image_code_id')
    # 2.校验参数是否齐全,正确
    if not all([mobile, image_code_client, image_code_id]):
        return jsonify(errno=response_code.RET.PARAMERR, errmsg='缺少参数')
    if not re.match(r'^1[345678][0-9]{9}$', mobile):
        return jsonify(errno=response_code.RET.PARAMERR, errmsg='手机号格式缺少参数')
    # 3.查询服务器储存的图片验证码
    try:
        image_code_server = redis_store.get('ImageCode:' + image_code_id)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=response_code.RET.DBERR, errmsg='查询图片验证码失败')
    if not image_code_server:
        return jsonify(errno=response_code.RET.NODATA, errmsg='查询图片验证码不存在')
    #     4.跟客户端传入图片验证码对比
    if image_code_server.lower() != image_code_client.lower():
        return jsonify(errno=response_code.RET.PARAMERR, errmsg='输入验证码有误')
    # 5.如果对比成功, 生成短信验证码, 并发送短信
    # '06':若果不够6位,用0补全.下面代码注释掉,不生产短信验证码
    sms_code = '%06d' % random.randint(0, 999999)
    current_app.logger.debug(sms_code)
    # result = CCP().send_template_sms(mobile, [sms_code, 5], 1)
    #
    # if result != 0:
    #     return jsonify(errno=response_code.RET.THIRDERR, errmsg='发送短信验证码失败')

    # 6.存储短信验证码到redis, 方便注册时比较
    try:
        redis_store.set('SMS:' + mobile, sms_code, constants.SMS_CODE_REDIS_EXPIRES)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=response_code.RET.DBERR, errmsg='保存验证码失败')
    # 7. 响应短信验证码发送的结果
    return jsonify(errno=response_code.RET.OK, errmsg='发送短信验证码成功')


@passport_blue.route('/image_code', methods=['GET'])
def image_code():
    '''提供图片验证码
    1.接受参数uuid
    2.效验参数(判断是否uuid为空)
    3.生成图片验证码
    4.保存图片验证码到redis
    5.修改image的Content-Type  = 'image/jpg'
    6.响应图片验证码
    '''
    #   1.获取参数
    imageCodeId = request.args.get('imageCodeId')
    #     2.校验参数
    if not imageCodeId:
        abort(403)
    #     3.生成图片验证码
    name, text, image = captcha.generate_captcha()
    current_app.logger.debug(text)
    #     4.保存图片验证码到redis
    try:
        redis_store.set('ImageCode:' + imageCodeId, text, constants.IMAGE_CODE_REDIS_EXPIRES)

    except Exception as e:
        current_app.logger.error(e)
        abort(500)
    # 5.修改Content-Type
    response = make_response(image)
    response.headers['Content-Type'] = 'image/jpg'

    #   6.响应图片验证码
    return response
